package com.nuanshui.heatedloan.extension.hibernate.type;

import java.io.Serializable;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Types;
import java.util.HashMap;
import java.util.Map;

import org.hibernate.HibernateException;
import org.hibernate.engine.spi.SessionImplementor;
import org.hibernate.usertype.UserType;

import com.alibaba.fastjson.JSON;

/**
 * Json字符串---->Map Map----->Json字符串
 * 参考http://jinnianshilongnian.iteye.com/blog/1497791 User: Zhang Kaitao Date:
 * 11-7-16 Time: 下午5:45
 */
public class JsonMapUserType implements UserType, Serializable {

	// private static ObjectMapper objectMapper = new ObjectMapper();

	static {
		// objectMapper.enableDefaultTyping();
	}

	@Override
	public int[] sqlTypes() {
		return new int[] { Types.VARCHAR };
	}

	@Override
	public Class returnedClass() {
		return JsonMap.class;
	}

	@Override
	public boolean equals(final Object o, final Object o1)
			throws HibernateException {
		if (o == o1) {
			return true;
		}
		if (o == null || o == null) {
			return false;
		}

		return o.equals(o1);
	}

	@Override
	public int hashCode(final Object o) throws HibernateException {
		return o.hashCode();
	}

	/**
	 * 从JDBC ResultSet读取数据,将其转换为自定义类型后返回 (此方法要求对克能出现null值进行处理)
	 * names中包含了当前自定义类型的映射字段名称
	 * 
	 * @param names
	 * @param owner
	 * @return
	 * @throws HibernateException
	 * @throws SQLException
	 */
	@Override
	public Object nullSafeGet(final ResultSet rs, final String[] names,
			final SessionImplementor session, final Object owner)
			throws HibernateException, SQLException {
		String json = rs.getString(names[0]);
		// try {
		// Map<Object, Object> map = objectMapper.readValue(json,
		// HashMap.class);
		// return new JsonMap(map);
		// } catch (IOException e) {
		// throw new HibernateException(e);
		// }
		Map<Object, Object> map = JSON.parseObject(json, HashMap.class);
		return new JsonMap(map);
	}

	/**
	 * 本方法将在Hibernate进行数据保存时被调用 我们可以通过PreparedStateme将自定义数据写入到对应的数据库表字段
	 */
	@Override
	public void nullSafeSet(final PreparedStatement st, final Object value,
			final int index, final SessionImplementor session)
			throws HibernateException, SQLException {
		if (value == null) {
			st.setNull(index, Types.VARCHAR);
		} else {
			// try {
			// st.setString(index, objectMapper.writeValueAsString((((JsonMap)
			// value).getMap())));
			// } catch (JsonProcessingException e) {
			// throw new HibernateException(e);
			// }
			st.setString(index, JSON.toJSONString((((JsonMap) value).getMap())));
		}
	}

	/**
	 * 提供自定义类型的完全复制方法 本方法将用构造返回对象 当nullSafeGet方法调用之后，我们获得了自定义数据对象，在向用户返回自定义数据之前，
	 * deepCopy方法将被调用，它将根据自定义数据对象构造一个完全拷贝，并将此拷贝返回给用户
	 * 此时我们就得到了自定义数据对象的两个版本，第一个是从数据库读出的原始版本，其二是我们通过
	 * deepCopy方法构造的复制版本，原始的版本将有Hibernate维护，复制版由用户使用。原始版本用作
	 * 稍后的脏数据检查依据；Hibernate将在脏数据检查过程中将两个版本的数据进行对比（通过调用
	 * equals方法），如果数据发生了变化（equals方法返回false），则执行对应的持久化操作
	 * 
	 * @param o
	 * @return
	 * @throws HibernateException
	 */
	@Override
	public Object deepCopy(final Object o) throws HibernateException {
		if (o == null)
			return null;
		JsonMap map = new JsonMap();
		map.setMap(((JsonMap) o).getMap());
		return map;
	}

	/**
	 * 本类型实例是否可变
	 * 
	 * @return
	 */
	@Override
	public boolean isMutable() {
		return true;
	}

	/* 序列化 */
	@Override
	public Serializable disassemble(final Object value)
			throws HibernateException {
		return ((Serializable) value);
	}

	/* 反序列化 */
	@Override
	public Object assemble(final Serializable cached, final Object owner)
			throws HibernateException {
		return cached;
	}

	@Override
	public Object replace(final Object original, final Object target,
			final Object owner) throws HibernateException {
		return original;
	}

}
